home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Modes / latex Mode / latexMacros.tcl < prev    next >
Encoding:
Text File  |  1998-12-08  |  41.8 KB  |  1,547 lines  |  [TEXT/ALFA]

  1. #############################################################################
  2. #############################################################################
  3. #
  4. # latexMacros.tcl (called from latex.tcl)
  5. #
  6. #############################################################################
  7. #
  8. # Author:  Tom Scavo <trscavo@syr.edu>, Vince Darley <darley@fas.harvard.edu>
  9. #
  10. #############################################################################
  11. #############################################################################
  12. proc latexMacros.tcl {} {}
  13. #############################################################################
  14. # Basic Commands
  15. #
  16. #############################################################################
  17. # ◊◊◊◊ label returners ◊◊◊◊ #
  18. proc TeX::labelDelim {} {
  19.     global TeXmodeVars
  20.     return $TeXmodeVars(standardTeXLabelDelimiter)
  21. }
  22. proc TeX::label {type} {
  23.     global TeXmodeVars
  24.     return "\\label\{${type}$TeXmodeVars(standardTeXLabelDelimiter)••\}"
  25. }
  26.  
  27. #--------------------------------------------------------------------------
  28. # ◊◊◊◊ Utilities: ◊◊◊◊
  29. #--------------------------------------------------------------------------
  30. # A keyboard-bound method of accessing menu commands.  Takes a list of 
  31. # menu items (i.e., the tail of a 'menu' command), the menu name (the 
  32. # argument of the '-n' switch) , and the name of a menu filter (the 
  33. # argument of the '-p' switch) as input, and displays these items in a 
  34. # list box.  If the chosen item is a menu command (as opposed to a 
  35. # submenu), it is passed to the menu filter; otherwise, 'chooseCommand' 
  36. # recursively calls itself until a menu command is chosen or the cancel 
  37. # button is pressed.
  38. #
  39. proc chooseCommand {menuItems {menuName ""} {menuFilterProc ""} {level 1}} {
  40.     watchCursor
  41.     if { $menuItems == "" } { return }
  42.     # Preprocess the list of menu items:
  43.     foreach    item $menuItems {
  44.         regsub -all {[<!/].} $item {} item
  45.         regsub -all {…}    $item {} item
  46.         lappend    menOut $item
  47.         if { [string match "menu*" $item] } {
  48.             if { [set ind [lsearch $item {-n}]] >= 0 } {
  49.                 lappend    top "[lindex $item [incr    ind]]:"
  50.             }
  51.         } elseif { ![string match "(*" $item] } {
  52.             lappend    top $item
  53.         }
  54.     }
  55.     # Present the menu items to the user:
  56.     set res    [listpick -p "Choose menu command (level $level):" $top]
  57.     # Either execute a command or recurse on a submenu:
  58.     if { [lsearch $menOut $res] >= 0 } {
  59.         # Execute the command via the menu filter, if necessary:
  60.         if { $menuFilterProc == "" } {
  61.             $res
  62.         } else {
  63.             $menuFilterProc $menuName $res
  64.         }
  65.     } else {
  66.         set res [string trimright $res {:}]
  67.         foreach    item $menOut {
  68.             if { [lsearch $item $res] >= 0 } {
  69.                 set menuItems [lindex $item end]
  70.                 # Determine the name of this submenu:
  71.                 if { [set ind [lsearch $item {-n}]] >= 0 } {
  72.                     set menuName [lindex $item [incr ind]]
  73.                 } else {
  74.                     set menuName ""
  75.                 }
  76.                 # Determine the name of the menu filter for this submenu:
  77.                 if { [set ind [lsearch $item {-p}]] >= 0 } {
  78.                     set menuFilterProc [lindex $item [incr ind]]
  79.                 } else {
  80.                     set menuFilterProc ""
  81.                 }
  82.                 return [chooseCommand $menuItems $menuName $menuFilterProc [incr level]]
  83.             }
  84.         }
  85.     }
  86. }
  87. # A keyboard-bound method of inserting environments.
  88. #
  89. proc chooseEnvironment {} {
  90.     set env [getEnvironment]
  91.     if { [catch {eval $env}] } {
  92.         doWrapEnvironment $env
  93.     }
  94. }
  95. proc getEnvironment {} {
  96.     if { [catch {prompt "Choose environment:" "abstract" "" \
  97.                  "abstract" \
  98.                  "array" \
  99.                  "center" \
  100.                  "description" \
  101.                  "displaymath" \
  102.                  "document" \
  103.                  "enumerate" \
  104.                  "eqnarray" \
  105.                  "eqnarray*" \
  106.                  "equation" \
  107.                  "figure" \
  108.                  "filecontents" \
  109.                  "flushleft" \
  110.                  "flushright" \
  111.                  "itemize" \
  112.                  "math" \
  113.                  "minipage" \
  114.                  "note" \
  115.                  "overlay" \
  116.                  "quotation" \
  117.                  "quote" \
  118.                  "slide" \
  119.                  "table" \
  120.                  "tabular" \
  121.                  "thebibliography" \
  122.                  "titlepage" \
  123.                  "verbatim" \
  124.                  "verse" \
  125.                 } env] } {
  126.         set env ""
  127.     }
  128.     return $env
  129. }
  130.  
  131. #############################################################################
  132. # Paragraph Mode Macros
  133. #
  134. #############################################################################
  135. #--------------------------------------------------------------------------
  136. # ◊◊◊◊ Documents: ◊◊◊◊
  137. #--------------------------------------------------------------------------
  138. proc newLaTeXDocument {} {
  139.     catch {prompt "Choose a documentclass:" "article" "" "article" \
  140.             "report" "book" "letter" "slides"} documentType
  141.     if {$documentType != "cancel"} {
  142.         new -m TeX
  143.         if { [catch {${documentType}Documentclass}] } {
  144.             wrapDocument "$documentType" 
  145.         }
  146.         while { [options] } {}
  147.         message "enter option (or leave blank)"
  148.     }
  149. }
  150. proc letterDocumentclass {} {
  151.     set    preamble "\r\\address\{%\r"
  152.     append preamble "    ••    \\\\    % insert your name here\r"
  153.     append preamble "    ••    \\\\    % insert your address here\r"
  154.     append preamble "    ••    \\\\    % insert more address here\r"
  155.     append preamble "    ••          % insert city-state-zip here\r"
  156.     append preamble "\}\r\r"
  157.     append preamble "\\date\{••\}  % optional\r"
  158.     append preamble "\\signature\{••\}\r\r"
  159.     set    body     "\r\\begin\{letter\}\{%\r"
  160.     append body     "    ••    \\\\    % insert addressee's name here\r"
  161.     append body     "    ••    \\\\    % insert addressee's address here\r"
  162.     append body     "    ••    \\\\    % insert more address here\r"
  163.     append body     "    ••          % insert addressee's city-state-zip here\r"
  164.     append body     "\}\r\r"
  165.     append body     "\\opening\{Dear ••,\}\r\r"
  166.     if {[isEmptyFile]} {
  167.         append body "% BODY OF LETTER\r"
  168.         append body "••\r\r"
  169.     } else {
  170.         if {[isSelectionAll]} {
  171.             set text [getSelect]
  172. #             deleteText 0 [maxPos]
  173.             append body "$text\r"
  174.         } else {
  175.             alertnote "nonempty file:  delete text or \'Select All\'\
  176.                 from the Edit menu"
  177.             return
  178.         }
  179.     }
  180.     append body "\\closing\{Sincerely,\}\r\r"
  181.     append body "\\encl\{••\}\r"
  182.     append body "\\cc\{••\}\r\r"
  183.     append body "\\end\{letter\}\r\r"
  184.     insertDocument "letter" $preamble $body
  185.     message "enter option (or leave blank)"
  186. }
  187. proc articleDocumentclass {} {
  188.     if { [wrapDocument "article"] } {
  189.         message "enter option (or leave blank)"
  190.     }
  191. }
  192. proc reportDocumentclass {} {
  193.     if { [wrapDocument "report"] } {
  194.         message "enter option (or leave blank)"
  195.     }
  196. }
  197. proc bookDocumentclass {} {
  198.     if { [wrapDocument "book"] } {
  199.         message "enter option (or leave blank)"
  200.     }
  201. }
  202. proc slidesDocumentclass {} {
  203.     if { [wrapDocument "slides"] } {
  204.         message "enter option (or leave blank)"
  205.     }
  206. }
  207. proc otherDocumentclass {} {
  208.     catch {prompt "What documentclass?" "article"} documentType
  209.     if {$documentType != "cancel"} {
  210.         if { [wrapDocument "$documentType"] } {
  211.             message "enter option (or leave blank)"
  212.         }
  213.     }
  214. }
  215. # If an option is inserted, return true; otherwise, return false.
  216. proc options {} {
  217.     set option [getOption]
  218.     if {$option != ""} {
  219.         insertOption $option
  220.         return 1
  221.     }
  222.     return 0
  223. }
  224. proc getOption {} {
  225.     catch {prompt "Choose an option:" "11pt" "" "10pt" "11pt" "12pt" "(-" \
  226.             "letterpaper" "legalpaper" "executivepaper" "a4paper" "a5paper" \
  227.             "b5paper" "(-" "landscape" "(-" "final" "draft" "(-" \
  228.             "oneside" "twoside" "(-" "openright" "openany" "(-" \
  229.             "onecolumn" "twocolumn" "(-" "notitlepage" "titlepage" \
  230.                   "openbib" "(-" "leqno" "(-" "fleqn"} optionName
  231.     if {$optionName != "cancel"} {
  232.         return $optionName
  233.     } else {
  234.         return ""
  235.     }
  236. }
  237. proc insertOption {option} {
  238.     global TeXmodeVars
  239.     set searchString {\\documentclass}
  240.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  241.     if {[llength $searchResult] == 0} {
  242.         if { $TeXmodeVars(searchNoisily) } {beep}
  243.         message "can\'t find \\documentclass"
  244.     } else {
  245.         set nextCharPos [lindex $searchResult 1]
  246.         goto $nextCharPos
  247.         set nextChar [lookAt $nextCharPos]
  248.         if {$nextChar == "\["} {
  249.             forwardChar
  250.             insertText $option
  251.             if {[lookAt [getPos]] != "\]"} {
  252.                 insertText ","
  253.             }
  254.         } elseif {$nextChar == "\{"} {
  255.             insertText "\[$option\]"
  256.         } else {
  257.             alertnote "unrecognizable \\documentclass statement"
  258.         }
  259.     }
  260. }
  261. proc insertPackage {package} {
  262.     global TeXmodeVars
  263.     # Check to see if $package is already loaded:
  264.     if { $package != "" } {
  265.         append searchString {^[^%]*\\usepackage\{.*} $package {.*\}}
  266.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  267.         if {[llength $searchResult] != 0} {
  268.             if { $TeXmodeVars(searchNoisily) } {beep}
  269.             message "$package package already loaded"
  270.             return
  271.         }
  272.     }
  273.     # Newlines are allowed in the arguments of \documentclass:
  274.     set searchString {\\documentclass(\[[^][]*\])?{[^{}]*}}
  275.     # Search for \documentclass command:
  276.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  277.     if {[llength $searchResult] == 0} {
  278.         if { $TeXmodeVars(searchNoisily) } {beep}
  279.         message "can't find \\documentclass"
  280.     } else {
  281.         pushPosition
  282.         goto [lindex $searchResult 1]
  283.         set txt "\r\\usepackage\{$package\}"
  284.         insertText $txt
  285.         backwardChar
  286.         message "Press <Ctl .> to return to previous position"
  287.     }
  288. }
  289. proc filecontents {} {
  290.     global searchNoisily
  291.     set searchString {\\documentclass}
  292.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  293.     if {[llength $searchResult] == 0} {
  294.         if {$searchNoisily} {beep}
  295.         message "can\'t find \\documentclass"
  296.         return
  297.     } else {
  298.         set prompt "File to be included:"
  299.         if {[catch {getfile $prompt} path]} {
  300.             return
  301.         } else {
  302.             replaceText [minPos] [minPos] [buildFilecontents $path]
  303.             goto [minPos]
  304.             message "file included"
  305.         }
  306.     }
  307. }
  308. proc filecontentsAll {} {
  309.     global searchNoisily
  310.     watchCursor
  311.     message "locating all input files…"
  312.     set currentWin [win::Current]
  313.     # Is the current window part of TeX fileset?
  314.     set fset [isWindowInFileset $currentWin "tex"]
  315.     if { $fset == "" } {
  316.         set searchString {\\documentclass}
  317.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  318.         if {[llength $searchResult] == 0} {
  319.             if {$searchNoisily} {beep}
  320.             message "can\'t find \\documentclass"
  321.             return
  322.         } else {
  323.             set text [getText [minPos] [maxPos]]
  324.         }
  325.     } else {
  326.         # Will not handle a base file that is open and dirty:
  327.         set text [buildFilecontents [texFilesetBaseName $fset]]
  328.     }
  329.     set currentDir [file dirname $currentWin]
  330.     set newText [texResolveAll $text $currentDir]
  331.     if { [string length $text] == [string length $newText] } {
  332.         beep
  333.         message "no files to include"
  334.     } else {
  335.         replaceText [minPos] [maxPos] $newText
  336.         goto [minPos]
  337.         message "all files included"
  338.     }
  339. }
  340. # Takes a LaTeX document string and a path as input, and returns
  341. # a modified document string with all filecontents environments
  342. # prepended.
  343. proc texResolveAll {latexDoc currentDir} {
  344.     global TeXmodeVars
  345.     set pairs [list \
  346.                 {{\\documentclass} {.cls}} {{\\LoadClass} {.cls}} \
  347.                 {{\\include} {.tex}} \
  348.                 {{\\usepackage} {.sty}} {{\\RequirePackage} {.sty}} \
  349.                 {{\\input} {}} \
  350.                 {{\\bibliography} {.bib}} {{\\bibliographystyle} {.bst}} \
  351.               ]
  352.     foreach macro $TeXmodeVars(boxMacroNames) {
  353.         regsub {\*} $macro {\\*} macro
  354.         lappend pairs [list \\\\$macro {}]
  355.     }
  356.     foreach pair $pairs {
  357.         set cmd [car $pair]
  358.         set ext [cadr $pair]
  359.         set searchString $cmd
  360.         append searchString {(\[[^][]*\])?{([^{}]*)}}
  361.         set searchText $latexDoc
  362.         while { [regexp -indices $searchString $searchText mtch dummy theArgs] } {
  363.             set begPos [lindex $theArgs 0]
  364.             set endPos [lindex $theArgs 1]
  365.             set args [string range $searchText $begPos $endPos]
  366.             foreach arg [split $args ,] {
  367.                 if { $cmd == {\\input} && ![string length [file extension $arg]] } {
  368.                     set ext {.tex}
  369.                 }
  370.                 set files [glob -nocomplain [file join $currentDir $arg*]]
  371.                 set filename [file join $currentDir $arg$ext]
  372.                 if { [lsearch -exact $files $filename] > -1 } {
  373.                     set tempDoc $latexDoc
  374.                     set latexDoc [buildFilecontents $filename]
  375.                     append latexDoc $tempDoc
  376.                 }
  377.             }
  378.             set searchText [string range $searchText [expr $endPos + 2] end]
  379.         }
  380.     }
  381.     return $latexDoc
  382. }
  383. # Takes a filename as input and returns a filecontents environment 
  384. # based on the contents of that file.  If a second argument is given,
  385. # use that as the argument of the filecontents environment instead
  386. # of the original filename.
  387. proc buildFilecontents {filename {newFilename {}}} {
  388.     set text [readFile $filename]
  389.     # Fix end-of-line characters:
  390.     regsub -all "\xa" $text "\xd" text
  391.     set envName "filecontents"
  392.     if { $newFilename == {} } {
  393.         set envArg "{[file tail $filename]}"
  394.     } else {
  395.         set envArg "{$newFilename}"
  396.     }
  397.     return [buildEnvironment $envName $envArg "$text\r" "\r\r"]
  398. }
  399. #--------------------------------------------------------------------------
  400. # ◊◊◊◊ Page Layout: ◊◊◊◊
  401. #--------------------------------------------------------------------------
  402. proc maketitle {} {
  403.     global searchNoisily
  404.     set searchString {\\document(class|style)(\[.*\])?\{.*\}}
  405.     set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString 0]
  406.     if {[llength $searchResult] == 0} {
  407.         if {$searchNoisily} {beep}
  408.         message "can\'t find \\documentclass or \\documentstyle"
  409.     } else {
  410.         set searchPos [lindex $searchResult 1]
  411.         set searchString {\\begin\{document\}}
  412.         set searchResult [search -s -n -f 1 -m 0 -i 1 -r 1 $searchString $searchPos]
  413.         if {[llength $searchResult] == 0} {
  414.             if {$searchNoisily} {beep}
  415.             message "can\'t find \\begin\{document\}"
  416.         } else {
  417.             goto [lindex $searchResult 1]
  418.             set currentPos [getPos]
  419.             set txt "\r\r% Definition of title page:"
  420.             append txt "\r\\title\{"
  421.             append txt "\r\t•title•\r\}"
  422.             append txt "\r\\author\{"
  423.             append txt "\r\t••\t% insert author(s) here"
  424.             append txt "\r\}"
  425.             append txt "\r\\date\{••\}\t% optional"
  426.             append txt "\r\r\\maketitle"
  427.             elec::Insertion $txt
  428.         }
  429.     }
  430. }
  431. proc abstract {} { doWrapEnvironment "abstract" }
  432. proc titlepage {} { doWrapEnvironment "titlepage" }
  433. proc getPagestyle {} {
  434.     catch {prompt "Choose a pagestyle:" "plain" "" "plain" "empty" \
  435.                   "headings" "myheadings"} pagestyleName
  436.     if {$pagestyleName != "cancel"} {
  437.         return $pagestyleName
  438.     } else {
  439.         return ""
  440.     }
  441. }
  442. proc pagestyle {} {
  443.     set pagestyleName [getPagestyle]
  444.     if {$pagestyleName != ""} {
  445.     insertObject "[openingCarriageReturn]\\pagestyle\{$pagestyleName\}[closingCarriageReturn]"
  446.     }
  447. }
  448. proc thispagestyle {} {
  449.     set pagestyleName [getPagestyle]
  450.     if {$pagestyleName != ""} {
  451.     insertObject "[openingCarriageReturn]\\thispagestyle\{$pagestyleName\}[closingCarriageReturn]"
  452.     }
  453. }
  454.  
  455. proc pagenumbering {} {
  456.     set pagenumberingStyle [getPagenumberingStyle]
  457.     if {$pagenumberingStyle != ""} {
  458.     insertObject "[openingCarriageReturn]\\pagenumbering\{$pagenumberingStyle\}[closingCarriageReturn]"
  459.     }
  460. }
  461. proc getPagenumberingStyle {} {
  462.     catch {prompt "Choose a pagenumbering style:" "arabic" "" "arabic" \
  463.       "roman" "Roman" "alph" "Alph"} pagenumberingStyle
  464.     if {$pagenumberingStyle != "cancel"} {
  465.     return $pagenumberingStyle
  466.     } else {
  467.     return ""
  468.     }
  469. }
  470. proc twocolumn {} {
  471.     insertObject "[openingCarriageReturn]\\twocolumn[closingCarriageReturn]"
  472. }
  473. proc onecolumn {} {
  474.     insertObject "[openingCarriageReturn]\\onecolumn[closingCarriageReturn]"
  475. }
  476. #--------------------------------------------------------------------------
  477. # ◊◊◊◊ Sectioning: ◊◊◊◊
  478. #--------------------------------------------------------------------------
  479. proc sectioning {item} {
  480.     append left [openingCarriageReturn] "\\${item}\{"
  481.     append right "\}"
  482.     if [regexp (part|chapter) $item] {
  483.         append right "\\thispagestyle{empty}"
  484.     }
  485.     append right "••" [closingCarriageReturn]
  486.     if [elec::Wrap $left $right] {
  487.         message "don't forget the label <Ctl Opt L>"
  488.     } else {
  489.         message "type the [string trim $item {*}] name and don't forget the label <Ctl Opt L>"
  490.     }
  491. }
  492. proc appendix {} {
  493.     insertObject "[openingCarriageReturn]\\appendix[closingCarriageReturn]"
  494. }
  495. #--------------------------------------------------------------------------
  496. # ◊◊◊◊ Text Style: ◊◊◊◊
  497. #--------------------------------------------------------------------------
  498.  
  499. proc emph {} {
  500.     if {[elec::Wrap "\\emph{" "}••"]} {
  501.         message "selected text has been emphasized"
  502.     } else {
  503.         message "enter text to be emphasized"
  504.     }
  505. }
  506. proc textup {} {
  507.     if {[elec::Wrap "\\textup{" "}••"]} {
  508.         message "selected text has upright shape"
  509.     } else {
  510.         message "enter text to have upright shape"
  511.     }
  512. }
  513. proc textit {} {
  514.     if {[elec::Wrap "\\textit{" "}••"]} {
  515.         message "selected text has italic shape"
  516.     } else {
  517.         message "enter text to have italic shape"
  518.     }
  519. }
  520. proc textsl {} {
  521.     if {[elec::Wrap "\\textsl{" "}••"]} {
  522.         message "selected text has slanted shape"
  523.     } else {
  524.         message "enter text to have slanted shape"
  525.     }
  526. }
  527. proc textsc {} {
  528.     if {[elec::Wrap "\\textsc{" "}••"]} {
  529.         message "selected text has small caps shape"
  530.     } else {
  531.         message "enter text to have small caps shape"
  532.     }
  533. }
  534. proc textmd {} {
  535.     if {[elec::Wrap "\\textmd{" "}••"]} {
  536.         message "selected text has been set in medium series"
  537.     } else {
  538.         message "enter text to be set in medium series"
  539.     }
  540. }
  541. proc textbf {} {
  542.     if {[elec::Wrap "\\textbf{" "}••"]} {
  543.         message "selected text has been set in bold series"
  544.     } else {
  545.         message "enter text to be set in bold series"
  546.     }
  547. }
  548. proc textrm {} {
  549.     if {[elec::Wrap "\\textrm{" "}••"]} {
  550.         message "selected text has been set with roman family"
  551.     } else {
  552.         message "enter text to be set using roman family"
  553.     }
  554. }
  555. proc textsf {} {
  556.     if {[elec::Wrap "\\textsf{" "}••"]} {
  557.         message "selected text has been set with sans serif family"
  558.     } else {
  559.         message "enter text to be set using sans serif family"
  560.     }
  561. }
  562. proc texttt {} {
  563.     if {[elec::Wrap "\\texttt{" "}••"]} {
  564.         message "selected text has been set with typewriter family"
  565.     } else {
  566.         message "enter text to be set using typewriter family"
  567.     }
  568. }
  569. proc textnormal {} {
  570.     if {[elec::Wrap "\\textnormal{" "}••"]} {
  571.         message "selected text has been set with normal style"
  572.     } else {
  573.         message "enter text to be set using normal style"
  574.     }
  575. }
  576.  
  577. proc em {} {
  578.     if {[elec::Wrap "{\\em " "}••"]} {
  579.         message "emphasized text set"
  580.     } else {
  581.         message "enter text to be emphasized"
  582.     }
  583. }
  584. proc upshape {} {
  585.     if {[elec::Wrap "{\\upshape " "}••"]} {
  586.         message "text set in upright shape"
  587.     } else {
  588.         message "enter text to be set in upright shape"
  589.     }
  590. }
  591. proc itshape {} {
  592.     if {[elec::Wrap "{\\itshape " "}••"]} {
  593.         message "text set in italics shape"
  594.     } else {
  595.         message "enter text to be set in italics shape"
  596.     }
  597. }
  598. proc slshape {} {
  599.     if {[elec::Wrap "{\\slshape " "}••"]} {
  600.         message "text set in slanted shape"
  601.     } else {
  602.         message "enter text to be set in slanted shape"
  603.     }
  604. }
  605. proc scshape {} {
  606.     if {[elec::Wrap "{\\scshape " "}••"]} {
  607.         message "text set in small caps shape"
  608.     } else {
  609.         message "enter text to be set in small caps shape"
  610.     }
  611. }
  612. proc mdseries {} {
  613.     if {[elec::Wrap "{\\mdseries " "}••"]} {
  614.         message "text set in medium series"
  615.     } else {
  616.         message "enter text to be set in medium series"
  617.     }
  618. }
  619. proc bfseries {} {
  620.     if {[elec::Wrap "{\\bfseries " "}••"]} {
  621.         message "text set in bold series"
  622.     } else {
  623.         message "enter text to be set in bold series"
  624.     }
  625. }
  626. proc rmfamily {} {
  627.     if {[elec::Wrap "{\\rmfamily " "}••"]} {
  628.         message "text set in roman family"
  629.     } else {
  630.         message "enter text to be set in roman family"
  631.     }
  632. }
  633. proc sffamily {} {
  634.     if {[elec::Wrap "{\\sffamily " "}••"]} {
  635.         message "text set in sans serif family"
  636.     } else {
  637.         message "enter text to be set in sans serif family"
  638.     }
  639. }
  640. proc ttfamily {} {
  641.     if {[elec::Wrap "{\\ttfamily " "}••"]} {
  642.         message "text set in typewriter family"
  643.     } else {
  644.         message "enter text to be set in typewriter family"
  645.     }
  646. }
  647. proc normalfont {} {
  648.     if {[elec::Wrap "{\\normalfont " "}••"]} {
  649.         message "text set in normal style"
  650.     } else {
  651.         message "enter text to be set in normal style"
  652.     }
  653. }
  654.  
  655. #--------------------------------------------------------------------------
  656. # ◊◊◊◊ Text Size: ◊◊◊◊
  657. #--------------------------------------------------------------------------
  658.  
  659. proc doTextSize {textSize} {
  660.     if {[elec::Wrap "{\\$textSize " "}••"]} {
  661.         message "$textSize text set"
  662.     } else {
  663.         message "enter $textSize text"
  664.     }
  665. }
  666.  
  667. #--------------------------------------------------------------------------
  668. # ◊◊◊◊ Text Commands: ◊◊◊◊
  669. #--------------------------------------------------------------------------
  670.  
  671. proc textsuperscript {} {
  672.     if {[elec::Wrap "\\textsuperscript{" "}••"]} {
  673.         message "text superscripted"
  674.     } else {
  675.         message "enter superscripted text"
  676.     }
  677. }
  678. proc textcircled {} {
  679.     if {[elec::Wrap "\\textcircled{" "}••"]} {
  680.         message "text circled"
  681.     } else {
  682.         message "enter circled text"
  683.     }
  684. }
  685.  
  686. #--------------------------------------------------------------------------
  687. # ◊◊◊◊ International: ◊◊◊◊
  688. #--------------------------------------------------------------------------
  689.  
  690. proc {ò} {} {
  691.     if {[elec::Wrap "\\`{" "}••"]} {
  692.         message "accent set"
  693.     } else {
  694.         message "enter single character"
  695.     }
  696. }
  697. proc {ó} {} {
  698.     if {[elec::Wrap "\\'{" "}••"]} {
  699.         message "accent set"
  700.     } else {
  701.         message "enter single character"
  702.     }
  703. }
  704. proc {ô} {} {
  705.     if {[elec::Wrap "\\^{" "}••"]} {
  706.         message "accent set"
  707.     } else {
  708.         message "enter single character"
  709.     }
  710. }
  711. proc {ö} {} {
  712.     if {[elec::Wrap "\\\"{" "}••"]} {
  713.         message "accent set"
  714.     } else {
  715.         message "enter single character"
  716.     }
  717. }
  718. proc {õ} {} {
  719.     if {[elec::Wrap "\\~{" "}••"]} {
  720.         message "accent set"
  721.     } else {
  722.         message "enter single character"
  723.     }
  724. }
  725. proc {ç} {} {insertObject "\\c\{c\}"}
  726. proc {Ç} {} {insertObject "\\c\{C\}"}
  727. proc {œ} {} {insertObject "\\oe"}
  728. proc {Œ} {} {insertObject "\\OE"}
  729. proc {æ} {} {insertObject "\\ae"}
  730. proc {Æ} {} {insertObject "\\AE"}
  731. proc {å} {} {insertObject "\\aa"}
  732. proc {Å} {} {insertObject "\\AA"}
  733. proc {ø} {} {insertObject "\\o"}
  734. proc {Ø} {} {insertObject "\\O"}
  735. proc {ss} {} {insertObject "\\ss"}
  736. proc {SS} {} {insertObject "\\SS"}
  737. proc {¿} {} {insertObject "?`"}
  738. proc {¡} {} {insertObject "!`"}
  739.  
  740.  
  741. #--------------------------------------------------------------------------
  742. # ◊◊◊◊ Boxes: ◊◊◊◊
  743. #--------------------------------------------------------------------------
  744.  
  745. proc mbox {} {
  746.     if {[elec::Wrap "\\mbox{" "}••"]} {
  747.         message "mbox set"
  748.     } else {
  749.         message "enter text"
  750.     }
  751. }
  752. proc makebox {} {
  753.     if {[elec::Wrap "\\makebox\[••\]\[••\]{" "}••"]} {
  754.         message "makebox set; enter the width and position"
  755.     } else {
  756.         message "enter the width and position of the makebox, then the text"
  757.     }
  758. }
  759. proc fbox {} {
  760.     if {[elec::Wrap "\\fbox{" "}••"]} {
  761.         message "fbox set"
  762.     } else {
  763.         message "enter text"
  764.     }
  765. }
  766. proc framebox {} {
  767.     if {[elec::Wrap "\\framebox\[••\]\[••\]{" "}••"]} {
  768.         message "framebox set; enter the width and position"
  769.     } else {
  770.         message "enter the width and position of the framebox, then the text"
  771.     }
  772. }
  773. proc newsavebox {} {
  774.     if {[elec::Wrap "\\newsavebox{" "}••"]} {
  775.         message "newsavebox defined"
  776.     } else {
  777.         message "enter the command name of the sbox or savebox"
  778.     }
  779. }
  780. proc sbox {} {
  781.     if {[elec::Wrap "\\sbox{••}{" "}••"]} {
  782.         message "sbox set; enter the command name"
  783.     } else {
  784.         message "enter the command name of the sbox, then the text"
  785.     }
  786. }
  787. proc savebox {} {
  788.     if {[elec::Wrap "\\savebox{••}\[••\]\[••\]{" "}••"]} {
  789.         message "savebox set; enter the command name"
  790.     } else {
  791.         message "enter the command name of the savebox"
  792.     }
  793. }
  794. proc usebox {} {
  795.     if {[elec::Wrap "\\usebox{" "}••"]} {
  796.         message "usebox declared"
  797.     } else {
  798.         message "enter the command name of the sbox or savebox"
  799.     }
  800. }
  801. proc raisebox {} {
  802.     if {[elec::Wrap "\\raisebox{••}\[••\]\[••\]{" "}••"]} {
  803.         message "raisebox set; enter the displacement"
  804.     } else {
  805.         message "enter the displacement of the raisebox"
  806.     }
  807. }
  808. proc parbox {} {
  809.     if {[elec::Wrap "\\parbox\[••\]\{••\}{" "}••"]} {
  810.         message "parbox set; enter the position and width"
  811.     } else {
  812.         message "enter the position \[b|c|t\] and width of the parbox, then the text"
  813.     }
  814. }
  815. proc rule {} {
  816.     insertObject "\\rule\[••\]\{••\}{••}••"
  817.     message "enter the displacement of the rule, then width and height"
  818. }
  819.  
  820. #--------------------------------------------------------------------------
  821. # ◊◊◊◊ Misc: ◊◊◊◊
  822. #--------------------------------------------------------------------------
  823.  
  824. proc verb {} {
  825.     if {[elec::Wrap "\\verb|" "|••"]} {
  826.         message "verbatim text set"
  827.     } else {
  828.         message "enter verbatim text"
  829.     }
  830. }
  831. proc footnote {} {
  832.     if {[elec::Wrap "\\footnote{" "}••"]} {
  833.         message "footnote set"
  834.     } else {
  835.         message "enter footnote"
  836.     }
  837. }
  838. proc marginalNote {} {
  839.     if {[elec::Wrap "\\marginpar{" "}••"]} {
  840.         message "marginal note set"
  841.     } else {
  842.         message "enter marginal note"
  843.     }
  844. }
  845.  
  846. proc insertLabel {} {
  847.     if {[elec::Wrap "\\label{" "}••"]} {
  848.         message "label defined"
  849.     } else {
  850.         message "enter label"
  851.     }
  852. }
  853. proc ref {} { 
  854.     global completion::in_progress_pos
  855.     if {[pos::compare [set completion::in_progress_pos] == [getPos]]} {
  856.     bind::Completion
  857.     } else {
  858.     if {[elec::Wrap "\\ref{" "}••" 1]} {
  859.         message "reference made"
  860.     } else {
  861.         message "enter reference label"
  862.     }
  863.     }
  864. }
  865. proc eqref {} { 
  866.     global completion::in_progress_pos standardTeXLabelDelimiter
  867.     if {[pos::compare [set completion::in_progress_pos] == [getPos]]} {
  868.     bind::Completion
  869.     } else {
  870.     if {[elec::Wrap "\\eqref\{eq${standardTeXLabelDelimiter}" "\}••" 1]} {
  871.         message "reference made"
  872.     } else {
  873.         message "enter reference label"
  874.     }
  875.     }
  876. }
  877. proc pageref {} { 
  878.     global completion::in_progress_pos
  879.     if {[pos::compare [set completion::in_progress_pos] == [getPos]]} {
  880.     bind::Completion
  881.     } else {
  882.     if {[elec::Wrap "\\pageref{" "}••" 1]} {
  883.         message "page reference made"
  884.     } else {
  885.         message "enter page reference label"
  886.     }
  887.     }
  888. }
  889. proc cite {} {
  890.     if {[elec::Wrap "\\cite{" "}••" 1]} {
  891.         message "citation made"
  892.     } else {
  893.         message "enter citation key"
  894.     }
  895. }
  896. proc nocite {} {
  897.     if {[elec::Wrap "\\nocite{" "}••"]} {
  898.         message "citation added to the list"
  899.     } else {
  900.         message "enter citation key"
  901.     }
  902. }
  903.  
  904. # Insert an \item or a \bibitem, depending on the context.
  905. proc insertItem {} {
  906.     set command [eval getText [searchEnvironment]]
  907.     set environment [extractCommandArg $command]
  908.     switch $environment {
  909.         "itemize" {
  910.             set text "\\item  ••"
  911.         }
  912.         "enumerate" {
  913.             set text "\\item  ••"
  914.         }
  915.         "description" {
  916.             set text "\\item\[••\]  ••"
  917.         }
  918.         "thebibliography" {
  919.             set text "\\bibitem{••}  ••"
  920.         }
  921.         default {
  922.             beep
  923.             message "insertItem: cursor in $environment environment"
  924.             return
  925.         }
  926.     }
  927.     set pos [getPos]
  928.     # Indentation should mirror that of an existing \item
  929.     # (if it exists)
  930.     elec::Insertion [openingCarriageReturn]$text
  931. }
  932.  
  933. proc quotes {} {
  934.     if {[elec::Wrap "`" "'••"]} {
  935.         message "text quoted"
  936.     } else {
  937.         message "enter text"
  938.     }
  939. }
  940. proc dblQuotes {} {
  941.     if {[elec::Wrap "``" "''••"]} {
  942.         message "text double quoted"
  943.     } else {
  944.         message "enter text"
  945.     }
  946. }
  947.  
  948. proc texLogo {} {insertObject "\\TeX"}
  949. proc latexLogo {} {insertObject "\\LaTeX"}
  950. proc latex2eLogo {} {insertObject "\\LaTeXe"}
  951. proc today {} {insertObject "\\today"}
  952.  
  953. proc dag {} {insertObject "\\dag"}
  954. proc ddag {} {insertObject "\\ddag"}
  955. proc sectionMark {} {insertObject "\\S"}
  956. proc paragraphMark {} {insertObject "\\P"}
  957. proc copyright {} {insertObject "\\copyright"}
  958. proc pounds {} {insertObject "\\pounds"}
  959.  
  960.  
  961. #############################################################################
  962. # Math Mode Macros
  963. #
  964. #############################################################################
  965.  
  966. #--------------------------------------------------------------------------
  967. # ◊◊◊◊ Math Modes: ◊◊◊◊
  968. #--------------------------------------------------------------------------
  969.  
  970. proc texMath {} {
  971.     checkMathMode "texMath" 0
  972.     if {[elec::Wrap "$" "$••"]} {
  973.         message "formula set"
  974.     } else {
  975.         message "enter formula"
  976.     }
  977. }
  978. proc texDisplaymath {} {
  979.     checkMathMode "texDisplaymath" 0
  980.     if {[elec::Wrap "$$" "$$••"]} {
  981.         message "displayed formula set"
  982.     } else {
  983.         message "enter displayed formula"
  984.     }
  985. }
  986. proc latexMath {} {
  987.     checkMathMode "latexMath" 0
  988.     if {[elec::Wrap "\\( " " \\)••"]} {
  989.         message "formula set"
  990.     } else {
  991.         message "enter formula"
  992.     }
  993. }
  994. proc latexDisplaymath {} {
  995.     checkMathMode "latexDisplaymath" 0
  996.     if {[elec::Wrap "\\\[ " " \\\]••"]} {
  997.         message "displayed formula set"
  998.     } else {
  999.         message "enter displayed formula"
  1000.     }
  1001. }
  1002.  
  1003. #--------------------------------------------------------------------------
  1004. # ◊◊◊◊ Math Style: ◊◊◊◊
  1005. #--------------------------------------------------------------------------
  1006.  
  1007. proc doMathStyle {mathStyle description} {
  1008.     checkMathMode "$mathStyle" 1
  1009.     if {[elec::Wrap "\\$mathStyle{" "}••"]} {
  1010.         message "selected text is $description"
  1011.     } else {
  1012.         message "enter text to be $description"
  1013.     }
  1014. }
  1015. proc doUppercaseMathStyle {mathStyle description} {
  1016.     checkMathMode "$mathStyle" 1
  1017.     # Allow upper-case alphabetic arguments only:
  1018.     if {[isSelection] && (![isUppercase] || ![isAlphabetic])} {
  1019.         beep
  1020.         alertnote "argument to \\$mathStyle must be UPPERCASE alphabetic"
  1021.         return
  1022.     }
  1023.     if {[elec::Wrap "\\$mathStyle{" "}••"]} {
  1024.         message "selected text is $description"
  1025.     } else {
  1026.         message "enter text to be $description (UPPERCASE letters only)"
  1027.     }
  1028. }
  1029.  
  1030.  
  1031. #--------------------------------------------------------------------------
  1032. # ◊◊◊◊ Formulas: ◊◊◊◊
  1033. #--------------------------------------------------------------------------
  1034.  
  1035. proc subscript {} {
  1036.     checkMathMode "subscript" 1
  1037.     if {[elec::Wrap "_{" "}••"]} {
  1038.         message "subscript set"
  1039.     } else {
  1040.         message "enter subscript"
  1041.     }
  1042. }
  1043. proc superscript {} {
  1044.     checkMathMode "superscript" 1
  1045.     if {[elec::Wrap "^{" "}••"]} {
  1046.         message "superscript set"
  1047.     } else {
  1048.         message "enter superscript"
  1049.     }
  1050. }
  1051. proc fraction {} {
  1052.     checkMathMode "fraction" 1
  1053.     set currentPos [getPos]
  1054.     if {[isSelection]} {
  1055.         set selection [getSelect]
  1056.         set args [split $selection /]
  1057.         set len [llength $args]
  1058.         deleteText $currentPos [selEnd]
  1059.         if {$len == 1} {
  1060.             insertObject "\\frac{$selection}{••}••"
  1061.             message "enter denominator"
  1062.         } else {
  1063.             set firstArg [lindex $args 0]
  1064.             set restArgs [lrange $args 1 [expr $len-1]]
  1065.             insertObject "\\frac{$firstArg}{[join $restArgs /]}"
  1066.             if {$len > 2} {message "beware of multiple /"}
  1067.         }
  1068.     } else {
  1069.         insertObject "\\frac{••}{••}••"
  1070.         message "enter numerator"
  1071.     }
  1072. }
  1073. proc squareRoot {} {
  1074.     checkMathMode "squareRoot" 1
  1075.     if {[elec::Wrap "\\sqrt{" "}••"]} {
  1076.         message "square root set"
  1077.     } else {
  1078.         message "enter formula"
  1079.     }
  1080. }
  1081. proc nthRoot {} {
  1082.     checkMathMode "nthRoot" 1
  1083.     if {[elec::Wrap "\\sqrt\[••\]{" "}••"]} {
  1084.         message "enter root"
  1085.     } else {
  1086.         message "enter root, then formula"
  1087.     }
  1088. }
  1089. proc oneParameter {} {
  1090.     checkMathMode "oneParameter" 1
  1091.     if {[elec::Wrap "\\••{" "}••"]} {
  1092.         message "enter command name"
  1093.     } else {
  1094.         message "enter command name, press <Tab>, enter argument"
  1095.     }
  1096. }
  1097. proc twoParameters {} {
  1098.     checkMathMode "twoParameters" 1
  1099.     if {[elec::Wrap "\\••{" "}{••}••"]} {
  1100.         message "enter command name"
  1101.     } else {
  1102.         message "enter command name, press <Tab>, enter argument, etc."
  1103.     }
  1104. }
  1105.  
  1106. #--------------------------------------------------------------------------
  1107. # ◊◊◊◊ Large Ops: ◊◊◊◊
  1108. #--------------------------------------------------------------------------
  1109.  
  1110. proc insertLargeOp {commandName} {
  1111.     checkMathMode "$commandName" 1
  1112.     insertObject "\\$commandName\_{••}^{••}••"
  1113. }
  1114. proc sum {} {insertLargeOp "sum"}
  1115. proc prod {} {insertLargeOp "prod"}
  1116. proc coprod {} {insertLargeOp "coprod"}
  1117. proc int {} {insertLargeOp "int"}
  1118. proc oint {} {insertLargeOp "oint"}
  1119. proc bigcap {} {insertLargeOp "bigcap"}
  1120. proc bigcup {} {insertLargeOp "bigcup"}
  1121. proc bigsqcup {} {insertLargeOp "bigsqcup"}
  1122. proc bigvee {} {insertLargeOp "bigvee"}
  1123. proc bigwedge {} {insertLargeOp "bigwedge"}
  1124. proc bigodot {} {insertLargeOp "bigodot"}
  1125. proc bigotimes {} {insertLargeOp "bigotimes"}
  1126. proc bigoplus {} {insertLargeOp "bigoplus"}
  1127. proc biguplus {} {insertLargeOp "biguplus"}
  1128.  
  1129. #--------------------------------------------------------------------------
  1130. # ◊◊◊◊ Delimiters: ◊◊◊◊
  1131. #--------------------------------------------------------------------------
  1132.  
  1133. proc delimitObject {leftDelim rightDelim} {
  1134.     if {[elec::Wrap $leftDelim $rightDelim]} {
  1135.         message "formula delimited"
  1136.     } else {
  1137.         message "enter formula"
  1138.     }
  1139. }
  1140. proc parentheses {} { checkMathMode "parentheses" 1; delimitObject "(" ")••" }
  1141. proc brackets {} { checkMathMode "brackets" 1; delimitObject "\[" "\]••" }
  1142. proc braces {} { checkMathMode "braces" 1; delimitObject "\\\{" "\\\}••" }
  1143. proc absoluteValue {} { checkMathMode "absoluteValue" 1; delimitObject "|" "|••" }
  1144. proc getDelims {} {
  1145.     catch {prompt "Choose delimiters:" "parentheses" "" "parentheses" \
  1146.                   "brackets" "braces" "angle brackets" "vertical bars" \
  1147.                   "double bars" "ceiling" "floor"} delimType
  1148.     if {$delimType != "cancel"} {
  1149.         switch $delimType {
  1150.             "parentheses" {
  1151.                 set leftDelim "("
  1152.                 set rightDelim ")"
  1153.             }
  1154.             "brackets" {
  1155.                 set leftDelim "\["
  1156.                 set rightDelim "\]"
  1157.             }
  1158.             "braces" {
  1159.                 set leftDelim "\\\{"
  1160.                 set rightDelim "\\\}"
  1161.             }
  1162.             "vertical bars" {
  1163.                 set leftDelim "|"
  1164.                 set rightDelim "|"
  1165.             }
  1166.             "double bars" {
  1167.                 set leftDelim "\\|"
  1168.                 set rightDelim "\\|"
  1169.             }
  1170.             "angle brackets" {
  1171.                 set leftDelim "\\langle"
  1172.                 set rightDelim "\\rangle"
  1173.             }
  1174.             "ceiling" {
  1175.                 set leftDelim "\\lceil"
  1176.                 set rightDelim "\\rceil"
  1177.             }
  1178.             "floor" {
  1179.                 set leftDelim "\\lfloor"
  1180.                 set rightDelim "\\rfloor"
  1181.             }
  1182.             default {
  1183.                 alertnote "\"$delimType\" not recognized"
  1184.                 return ""
  1185.             }
  1186.         }
  1187.         return [list $leftDelim $rightDelim]
  1188.     } else {return ""}
  1189. }
  1190. proc otherDelims {} {
  1191.     checkMathMode "otherDelims" 1
  1192.     set delims [getDelims]
  1193.     if {$delims != ""} {
  1194.         set leftDelim [lindex $delims 0]
  1195.         set rightDelim [lindex $delims 1]
  1196.         delimitObject "$leftDelim" "$rightDelim••"
  1197.     }
  1198. }
  1199. proc {half-openInterval} {} {
  1200.     checkMathMode "half-openInterval" 1; delimitObject "(" "\]••"
  1201. }
  1202. proc {half-closedInterval} {} {
  1203.     checkMathMode "half-closedInterval" 1; delimitObject "\[" ")••"
  1204. }
  1205. proc insertBigDelims {leftDelim rightDelim isMultiline} {
  1206.     checkMathMode "insertBigDelims" 1
  1207.     if {$isMultiline} {
  1208.         doWrapStructure $leftDelim "" $rightDelim
  1209.     } else {
  1210.         if { [elec::Wrap $leftDelim $rightDelim] } {
  1211.             message "formula delimited"
  1212.         } else {
  1213.             message "enter formula"
  1214.         }
  1215.     }
  1216. }
  1217. proc bigParens {} {
  1218.     checkMathMode "bigParens" 1; insertBigDelims "\\left(" "\\right)••" 0
  1219. }
  1220. proc multiBigParens {} {
  1221.     checkMathMode "multiBigParens" 1; insertBigDelims "\\left(" "\\right)••" 1
  1222. }
  1223. proc bigBrackets {} {
  1224.     checkMathMode "bigBrackets" 1; insertBigDelims "\\left\[" "\\right\]••" 0
  1225. }
  1226. proc multiBigBrackets {} {
  1227.     checkMathMode "multiBigBrackets" 1; insertBigDelims "\\left\[" "\\right\]••" 1
  1228. }
  1229. proc bigBraces {} {
  1230.     checkMathMode "bigBraces" 1; insertBigDelims "\\left\\\{" "\\right\\\}••" 0
  1231. }
  1232. proc multiBigBraces {} {
  1233.     checkMathMode "multiBigBraces" 1; insertBigDelims "\\left\\\{" "\\right\\\}••" 1
  1234. }
  1235. proc bigAbsValue {} {
  1236.     checkMathMode "bigAbsValue" 1; insertBigDelims "\\left|" "\\right|••" 0
  1237. }
  1238. proc multiBigAbsValue {} {
  1239.     checkMathMode "multiBigAbsValue" 1; insertBigDelims "\\left|" "\\right|••" 1
  1240. }
  1241. proc doOtherBigDelims {name isMultiline} {
  1242.     checkMathMode $name 1
  1243.     set delims [getDelims]
  1244.     if {$delims != ""} {
  1245.         append leftDelim "\\left" [lindex $delims 0]
  1246.         append rightDelim "\\right" [lindex $delims 1]
  1247.         insertBigDelims "$leftDelim" "$rightDelim••" $isMultiline
  1248.     }
  1249. }
  1250. proc otherBigDelims {} {
  1251.     doOtherBigDelims "otherBigDelims" 0
  1252. }
  1253. proc otherMultiBigDelims {} {
  1254.     doOtherBigDelims "otherMultiBigDelims" 1
  1255. }
  1256. proc bigLeftBrace {} {
  1257.     checkMathMode "bigLeftBrace" 1
  1258.     insertBigDelims "\\left\\\{" "\\right.••" 0
  1259. }
  1260. proc multiBigLeftBrace {} {
  1261.     checkMathMode "multiBigLeftBrace" 1
  1262.     insertBigDelims "\\left\\\{" "\\right.••" 1
  1263. }
  1264. proc doOtherMixedBigDelims {name isMultiline} {
  1265.     checkMathMode $name 1
  1266.     catch {prompt "Choose LEFT delimiter:" "parenthesis" "" "parenthesis" \
  1267.                   "bracket" "brace" "vertical bar" "double bar" \
  1268.                   "angle bracket" "ceiling" "floor" "slash" "backslash" \
  1269.                   "none"} delimType
  1270.     if {$delimType != "cancel"} {
  1271.         switch $delimType {
  1272.             "parenthesis" {set leftDelim "("}
  1273.             "bracket" {set leftDelim "\["}
  1274.             "brace" {set leftDelim "\\\{"}
  1275.             "vertical bar" {set leftDelim "|"}
  1276.             "double bar" {set leftDelim "\\|"}
  1277.             "angle bracket" {set leftDelim "\\langle"}
  1278.             "ceiling" {set leftDelim "\\lceil"}
  1279.             "floor" {set leftDelim "\\lfloor"}
  1280.             "slash" {set leftDelim "/"}
  1281.             "backslash" {set leftDelim "\\backslash"}
  1282.             "none" {set leftDelim "."}
  1283.             default {
  1284.                 alertnote "\"$delimType\" not recognized"
  1285.                 return
  1286.             }
  1287.         }
  1288.         catch {prompt "Choose RIGHT delimiter:" "parenthesis" "" "parenthesis" \
  1289.                       "bracket" "brace" "vertical bar" "double bar" \
  1290.                       "angle bracket" "ceiling" "floor" "slash" "backslash" \
  1291.                       "none"} delimType
  1292.         if {$delimType != "cancel"} {
  1293.             switch $delimType {
  1294.                 "parenthesis" {set rightDelim ")"}
  1295.                 "bracket" {set rightDelim "\]"}
  1296.                 "brace" {set rightDelim "\\\}"}
  1297.                 "vertical bar" {set rightDelim "|"}
  1298.                 "double bar" {set rightDelim "\\|"}
  1299.                 "angle bracket" {set rightDelim "\\rangle"}
  1300.                 "ceiling" {set rightDelim "\\rceil"}
  1301.                 "floor" {set rightDelim "\\rfloor"}
  1302.                 "slash" {set rightDelim "/"}
  1303.                 "backslash" {set rightDelim "\\backslash"}
  1304.                 "none" {set rightDelim "."}
  1305.                 default {
  1306.                     alertnote "\"$delimType\" not recognized"
  1307.                     return
  1308.                 }
  1309.             }
  1310.             insertBigDelims "\\left$leftDelim" "\\right$rightDelim••" $isMultiline
  1311.         }
  1312.     }
  1313. }
  1314. proc otherMixedBigDelims {} {
  1315.     doOtherMixedBigDelims "otherMixedBigDelims" 0
  1316. }
  1317. proc otherMultiMixedBigDelims {} {
  1318.     doOtherMixedBigDelims "otherMultiMixedBigDelims" 1
  1319. }
  1320. #--------------------------------------------------------------------------
  1321. # ◊◊◊◊ Accents: ◊◊◊◊
  1322. #--------------------------------------------------------------------------
  1323. proc acute {} {
  1324.     checkMathMode "acute" 1
  1325.     if {[isSelection] > 1} {
  1326.         alertnote "Warning: only a single character may be accented!"
  1327.     }
  1328.     if {[elec::Wrap "\\acute{" "}••"]} {
  1329.         message "accent set"
  1330.     } else {
  1331.         message "enter one character"
  1332.     }
  1333. }
  1334. proc bar {} {
  1335.     checkMathMode "bar" 1
  1336.     if {[isSelection] > 1} {
  1337.         alertnote "Warning: only a single character may be accented!"
  1338.     }
  1339.     if {[elec::Wrap "\\bar{" "}••"]} {
  1340.         message "accent set"
  1341.     } else {
  1342.         message "enter one character"
  1343.     }
  1344. }
  1345. proc breve {} {
  1346.     checkMathMode "breve" 1
  1347.     if {[isSelection] > 1} {
  1348.         alertnote "Warning: only a single character may be accented!"
  1349.     }
  1350.     if {[elec::Wrap "\\breve{" "}••"]} {
  1351.         message "accent set"
  1352.     } else {
  1353.         message "enter one character"
  1354.     }
  1355. }
  1356. proc check {} {
  1357.     checkMathMode "check" 1
  1358.     if {[isSelection] > 1} {
  1359.         alertnote "Warning: only a single character may be accented!"
  1360.     }
  1361.     if {[elec::Wrap "\\check{" "}••"]} {
  1362.         message "accent set"
  1363.     } else {
  1364.         message "enter one character"
  1365.     }
  1366. }
  1367. proc dot {} {
  1368.     checkMathMode "dot" 1
  1369.     if {[isSelection] > 1} {
  1370.         alertnote "Warning: only a single character may be accented!"
  1371.     }
  1372.     if {[elec::Wrap "\\dot{" "}••"]} {
  1373.         message "accent set"
  1374.     } else {
  1375.         message "enter one character"
  1376.     }
  1377. }
  1378. proc ddot {} {
  1379.     checkMathMode "ddot" 1
  1380.     if {[isSelection] > 1} {
  1381.         alertnote "Warning: only a single character may be accented!"
  1382.     }
  1383.     if {[elec::Wrap "\\ddot{" "}••"]} {
  1384.         message "accent set"
  1385.     } else {
  1386.         message "enter one character"
  1387.     }
  1388. }
  1389. proc grave {} {
  1390.     checkMathMode "grave" 1
  1391.     if {[isSelection] > 1} {
  1392.         alertnote "Warning: only a single character may be accented!"
  1393.     }
  1394.     if {[elec::Wrap "\\grave{" "}••"]} {
  1395.         message "accent set"
  1396.     } else {
  1397.         message "enter one character"
  1398.     }
  1399. }
  1400. proc hat {} {
  1401.     checkMathMode "hat" 1
  1402.     if {[isSelection] > 1} {
  1403.         alertnote "Warning: only a single character may be accented!"
  1404.     }
  1405.     if {[elec::Wrap "\\hat{" "}••"]} {
  1406.         message "accent set"
  1407.     } else {
  1408.         message "enter one character"
  1409.     }
  1410. }
  1411. proc tilde {} {
  1412.     checkMathMode "tilde" 1
  1413.     if {[isSelection] > 1} {
  1414.         alertnote "Warning: only a single character may be accented!"
  1415.     }
  1416.     if {[elec::Wrap "\\tilde{" "}••"]} {
  1417.         message "accent set"
  1418.     } else {
  1419.         message "enter one character"
  1420.     }
  1421. }
  1422. proc vec {} {
  1423.     checkMathMode "vec" 1
  1424.     if {[isSelection] > 1} {
  1425.         alertnote "Warning: only a single character may be accented!"
  1426.     }
  1427.     if {[elec::Wrap "\\vec{" "}••"]} {
  1428.         message "accent set"
  1429.     } else {
  1430.         message "enter one character"
  1431.     }
  1432. }
  1433. proc widehat {} {
  1434.     checkMathMode "widehat" 1
  1435.     if {[isSelection] > 3} {
  1436.         alertnote "Warning: only a few characters may be accented!"
  1437.     }
  1438.     if {[elec::Wrap "\\widehat{" "}••"]} {
  1439.         message "accent set"
  1440.     } else {
  1441.         message "enter a few characters"
  1442.     }
  1443. }
  1444. proc widetilde {} {
  1445.     checkMathMode "widetilde" 1
  1446.     if {[isSelection] > 3} {
  1447.         alertnote "Warning: only a few characters may be accented!"
  1448.     }
  1449.     if {[elec::Wrap "\\widetilde{" "}••"]} {
  1450.         message "accent set"
  1451.     } else {
  1452.         message "enter a few characters"
  1453.     }
  1454. }
  1455.  
  1456. #--------------------------------------------------------------------------
  1457. # ◊◊◊◊ Grouping: ◊◊◊◊
  1458. #--------------------------------------------------------------------------
  1459. proc underline {} {
  1460.     checkMathMode "underline" 1
  1461.     if {[elec::Wrap "\\underline{" "}••"]} {
  1462.         message "selection underlined"
  1463.     } else {
  1464.         message "enter text"
  1465.     }
  1466. }
  1467. proc overline {} {
  1468.     checkMathMode "overline" 1
  1469.     if {[elec::Wrap "\\overline{" "}••"]} {
  1470.         message "selection overlined"
  1471.     } else {
  1472.         message "enter text"
  1473.     }
  1474. }
  1475. proc underbrace {} {
  1476.     checkMathMode "underbrace" 1
  1477.     if {[elec::Wrap "\\underbrace{" "}••"]} {
  1478.         message "selection underbraced"
  1479.     } else {
  1480.         message "enter text"
  1481.     }
  1482. }
  1483. proc overbrace {} {
  1484.     checkMathMode "overbrace" 1
  1485.     if {[elec::Wrap "\\overbrace{" "}••"]} {
  1486.         message "selection overbraced"
  1487.     } else {
  1488.         message "enter text"
  1489.     }
  1490. }
  1491. proc overrightarrow {} {
  1492.     checkMathMode "overrightarrow" 1
  1493.     if {[elec::Wrap "\\overrightarrow{" "}••"]} {
  1494.         message "selection overrightarrowed"
  1495.     } else {
  1496.         message "enter text"
  1497.     }
  1498. }
  1499. proc overleftarrow {} {
  1500.     checkMathMode "overleftarrow" 1
  1501.     if {[elec::Wrap "\\overleftarrow{" "}••"]} {
  1502.         message "selection overleftarrowed"
  1503.     } else {
  1504.         message "enter text"
  1505.     }
  1506. }
  1507. proc stackrel {} {
  1508.     checkMathMode "stackrel" 1
  1509.     set currentPos [getPos]
  1510.     if {[insertObject "\\stackrel{••}{••}••"]} {
  1511.         message "1st arg scriptstyle"
  1512.     }
  1513. }
  1514. #--------------------------------------------------------------------------
  1515. # ◊◊◊◊ Spacing: ◊◊◊◊
  1516. #--------------------------------------------------------------------------
  1517. proc negThin {} {checkMathMode "negThin" 1; insertObject "\\!"}
  1518. proc thin {} {checkMathMode "thin" 1; insertObject "\\,"}
  1519. proc medium {} {checkMathMode "medium" 1; insertObject "\\:"}
  1520. proc thick {} {checkMathMode "thick" 1; insertObject "\\;"}
  1521. proc quad {} {checkMathMode "quad" 1; insertObject "\\quad"}
  1522. proc qquad {} {checkMathMode "qquad" 1; insertObject "\\qquad"}
  1523. proc hspace {} {
  1524.     checkMathMode "hspace" 1
  1525.     if {[elec::Wrap "\\hspace{" "}••"]} {
  1526.         message "spacing set"
  1527.     } else {
  1528.         message "enter the desired horizontal spacing"
  1529.     }
  1530. }
  1531. proc vspace {} {
  1532.     checkMathMode "vspace" 1
  1533.     if {[elec::Wrap "\\vspace{" "}••"]} {
  1534.         message "spacing set"
  1535.     } else {
  1536.         message "enter the desired horizontal spacing"
  1537.     }
  1538. }
  1539. proc hfill {} {checkMathMode "hfill" 1; insertObject "\\hfill"}
  1540. proc vfill {} {checkMathMode "vfill" 1; insertObject "\\vfill"}
  1541. proc smallskip {} {checkMathMode "smallskip" 1; insertObject "\\smallskip"}
  1542. proc medskip {} {checkMathMode "medskip" 1; insertObject "\\medskip"}
  1543. proc bigskip {} {checkMathMode "bigskip" 1; insertObject "\\bigskip"}
  1544.